<script setup lang="tsx">
import { onMounted, ref } from 'vue';
import { NButton, NDivider, NEllipsis, NImage, NTag, NTooltip } from 'naive-ui';
import { useBoolean, useLoading } from '@sa/hooks';
import { fetchBatchDeleteOss, fetchGetOssList } from '@/service/api/system/oss';
import { fetchGetConfigByKey, fetchUpdateConfigByKey } from '@/service/api/system/config';
import { useAppStore } from '@/store/modules/app';
import { useTable, useTableOperate } from '@/hooks/common/table';
import { useAuth } from '@/hooks/business/auth';
import { useDownload } from '@/hooks/business/download';
import { useRouterPush } from '@/hooks/common/router';
import { isImage } from '@/utils/common';
import { handleCopy } from '@/utils/copy';
import { $t } from '@/locales';
import ButtonIcon from '@/components/custom/button-icon.vue';
import OssSearch from './modules/oss-search.vue';
import OssUploadModal from './modules/oss-upload-modal.vue';
defineOptions({
  name: 'OssList'
});

const { routerPushByKey } = useRouterPush();
const { hasAuth } = useAuth();
const { oss } = useDownload();
const appStore = useAppStore();

const fileUploadType = ref<'file' | 'image'>('file');
const { bool: preview, setBool: setPreview } = useBoolean(true);
const { loading: previewLoading, startLoading: startPreviewLoading, endLoading: endPreviewLoading } = useLoading(false);
const { bool: uploadVisible, setTrue: showFUploadModal } = useBoolean(false);
const {
  columns,
  columnChecks,
  data,
  getData,
  getDataByPage,
  loading,
  mobilePagination,
  searchParams,
  resetSearchParams
} = useTable({
  apiFn: fetchGetOssList,
  apiParams: {
    pageNum: 1,
    pageSize: 10,
    // if you want to use the searchParams in Form, you need to define the following properties, and the value is null
    // the value can not be undefined, otherwise the property in Form will not be reactive
    fileName: null,
    originalName: null,
    fileSuffix: null,
    service: null,
    params: {}
  },
  columns: () => [
    {
      type: 'selection',
      align: 'center',
      width: 48
    },
    {
      key: 'index',
      title: $t('common.index'),
      align: 'center',
      width: 64
    },
    {
      key: 'ossId',
      title: '对象存储主键',
      align: 'center',
      minWidth: 120
    },
    {
      key: 'fileName',
      title: '文件名',
      align: 'center',
      ellipsis: {
        tooltip: true,
        lineClamp: 3
      },
      minWidth: 120
    },
    {
      key: 'originalName',
      title: '原名',
      align: 'center',
      ellipsis: {
        tooltip: true,
        lineClamp: 3
      },
      minWidth: 120
    },
    {
      key: 'fileSuffix',
      title: '文件后缀名',
      align: 'center',
      minWidth: 100
    },
    {
      key: 'url',
      title: 'URL地址',
      align: 'center',
      minWidth: 120,
      render: row => {
        if (preview.value && isImage(row.fileSuffix)) {
          return <NImage class="h-40px w-40px object-contain" src={row.url} />;
        }
        return (
          <NTooltip>
            {{
              default: () => <span>点击复制</span>,
              trigger: () => (
                <div class="cursor-pointer" onClick={async () => await handleCopy(row.url)}>
                  <NEllipsis line-clamp={3} tooltip={false}>
                    {row.url}
                  </NEllipsis>
                </div>
              )
            }}
          </NTooltip>
        );
      }
    },
    {
      key: 'createTime',
      title: '创建时间',
      align: 'center',
      minWidth: 120
    },
    {
      key: 'createByName',
      title: '上传人',
      align: 'center',
      minWidth: 120
    },
    {
      key: 'service',
      title: '服务商',
      align: 'center',
      minWidth: 100,
      render: row => {
        return <NTag type="primary">{row.service}</NTag>;
      }
    },
    {
      key: 'operate',
      title: $t('common.operate'),
      align: 'center',
      width: 130,
      render: row => {
        const divider = () => {
          if (!hasAuth('system:oss:download') || !hasAuth('system:oss:delete')) {
            return null;
          }
          return <NDivider vertical />;
        };

        const downloadBtn = () => {
          if (!hasAuth('system:oss:download')) {
            return null;
          }
          return (
            <ButtonIcon
              text
              type="primary"
              icon="material-symbols:download-rounded"
              class="text-20px"
              tooltipContent={$t('common.download')}
              onClick={() => download(row.ossId!)}
            />
          );
        };

        const deleteBtn = () => {
          if (!hasAuth('system:oss:delete')) {
            return null;
          }
          return (
            <ButtonIcon
              text
              type="error"
              icon="material-symbols:delete-outline"
              class="text-20px"
              tooltipContent={$t('common.delete')}
              popconfirmContent={$t('common.confirmDelete')}
              onPositiveClick={() => handleDelete(row.ossId!)}
            />
          );
        };

        return (
          <div class="flex-center gap-8px">
            {downloadBtn()}
            {divider()}
            {deleteBtn()}
          </div>
        );
      }
    }
  ]
});

const { handleAdd, checkedRowKeys, onBatchDeleted, onDeleted } = useTableOperate(data, getData);

async function handleBatchDelete() {
  // request
  const { error } = await fetchBatchDeleteOss(checkedRowKeys.value);
  if (error) return;
  onBatchDeleted();
}

async function handleDelete(ossId: CommonType.IdType) {
  // request
  const { error } = await fetchBatchDeleteOss([ossId]);
  if (error) return;
  onDeleted();
}

function download(ossId: CommonType.IdType) {
  oss(ossId);
}

function handleUpload(type: 'file' | 'image') {
  fileUploadType.value = type;
  showFUploadModal();
}

async function getConfigKey() {
  const { data: previewStr, error } = await fetchGetConfigByKey('sys.oss.previewListResource');
  if (error) return;
  setPreview(previewStr === 'true');
}

onMounted(() => {
  getConfigKey();
});

async function handleUpdatePreview(checked: boolean) {
  setPreview(!checked);
  window.$dialog?.warning({
    title: '提示',
    content: `是否确认${checked ? '开启' : '关闭'}预览？`,
    positiveText: '确认',
    negativeText: '取消',
    onPositiveClick: async () => {
      startPreviewLoading();
      const { error } = await fetchUpdateConfigByKey({
        configKey: 'sys.oss.previewListResource',
        configValue: String(checked)
      });
      if (error) {
        setPreview(!checked);
        endPreviewLoading();
        return;
      }
      setPreview(checked);
      window.$message?.success('更新成功');
      endPreviewLoading();
    },
    onNegativeClick: () => {
      setPreview(!checked);
    }
  });
}

function handleToOssConfig() {
  routerPushByKey('system_oss-config');
}
</script>

<template>
  <div class="min-h-500px flex-col-stretch gap-16px overflow-hidden lt-sm:overflow-auto">
    <OssSearch v-model:model="searchParams" @reset="resetSearchParams" @search="getDataByPage" />
    <NCard title="OSS 对象存储列表" :bordered="false" size="small" class="sm:flex-1-hidden card-wrapper">
      <template #header-extra>
        <TableHeaderOperation
          v-model:columns="columnChecks"
          :disabled-delete="checkedRowKeys.length === 0"
          :loading="loading"
          :show-add="false"
          :show-delete="hasAuth('system:oss:delete')"
          @add="handleAdd"
          @delete="handleBatchDelete"
          @refresh="getData"
        >
          <template #prefix>
            <NSwitch
              v-model:value="preview"
              class="mt-1px"
              :loading="previewLoading"
              size="large"
              :round="false"
              @update:value="handleUpdatePreview"
            >
              <template #checked>
                <span class="text-14px">禁用预览</span>
              </template>
              <template #unchecked>
                <span class="text-14px">开启预览</span>
              </template>
            </NSwitch>

            <NButton size="small" ghost @click="handleUpload('file')">
              <template #icon>
                <icon-material-symbols:upload-rounded />
              </template>
              上传文件
            </NButton>
            <NButton size="small" ghost @click="handleUpload('image')">
              <template #icon>
                <icon-material-symbols:image-outline />
              </template>
              上传图片
            </NButton>
            <NButton type="primary" size="small" ghost @click="handleToOssConfig">
              <template #icon>
                <icon-hugeicons:configuration-01 />
              </template>
              配置管理
            </NButton>
          </template>
        </TableHeaderOperation>
      </template>
      <NDataTable
        v-model:checked-row-keys="checkedRowKeys"
        :columns="columns"
        :data="data"
        size="small"
        :flex-height="!appStore.isMobile"
        :scroll-x="962"
        :loading="loading"
        remote
        :row-key="row => row.ossId"
        :pagination="mobilePagination"
        class="sm:h-full"
      />
      <OssUploadModal v-model:visible="uploadVisible" :upload-type="fileUploadType" @close="getData" />
    </NCard>
  </div>
</template>

<style scoped>
.n-switch {
  --n-rail-height: 27px !important;
}
</style>
